package com.zs.netty.jdknio;

import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.CharBuffer;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;
import java.nio.channels.SocketChannel;
import java.nio.charset.Charset;
import java.util.Iterator;
import java.util.Scanner;

/**
 * NIO客户端
 *
 * @author madison
 */
public class NIOClient {
    //通道管理器
    private Selector selector;
    private boolean isOver;

    /**
     * 获得一个Socket通道，并对该通道做一些初始化的工作
     *
     * @param ip   连接的服务器的ip
     * @param port 连接的服务器的端口号
     * @throws IOException
     */
    public void initClient(String ip, int port) throws IOException {
        // 获得一个通道管理器
        selector = Selector.open();
        // 获得一个Socket通道
        SocketChannel channel = SocketChannel.open();

        // 设置通道为非阻塞
        channel.configureBlocking(false);
        // 客户端连接服务器,其实方法执行并没有实现连接，需要在listen（）方法中调
        //用channel.finishConnect();才能完成连接
        channel.connect(new InetSocketAddress(ip, port));
        //将通道管理器和该通道绑定，并为该通道注册SelectionKey.OP_CONNECT事件。
        channel.register(selector, SelectionKey.OP_CONNECT);
    }

    /**
     * 采用轮询的方式监听selector上是否有需要处理的事件，如果有，则进行处理
     *
     * @throws IOException
     */
    public void listen() throws IOException {
        isOver = false;
        // 轮询访问selector
        while (!isOver) {
            selector.select();
            // 获得selector中选中的项的迭代器
            Iterator ite = selector.selectedKeys().iterator();
            while (ite.hasNext()) {
                SelectionKey key = (SelectionKey) ite.next();
                // 删除已选的key,以防重复处理
                ite.remove();
                if (key.isConnectable()) {  // 连接事件发生
                    connect(key);
                } else if (key.isReadable()) { // 获得了可读的事件
                    read(key);
                }
            }

        }
    }

    private void connect(SelectionKey key) throws IOException {
        SocketChannel channel = (SocketChannel) key.channel();
        // 如果正在连接，则完成连接
        if (channel.isConnectionPending()) {
            channel.finishConnect();
        }
        // 设置成非阻塞
        channel.configureBlocking(false);
        //在和服务端连接成功之后，为了可以接收到服务端的信息，需要给通道设置读的权限。
        channel.register(selector, SelectionKey.OP_READ);

        //在这里可以给服务端发送信息哦
        Scanner scanner = new Scanner(System.in);
        System.out.println("请输入：");
        String str = scanner.nextLine();
        ByteBuffer outBuffer = Charset.forName("UTF-8").newEncoder().encode(CharBuffer.wrap(str));
//                    channel.write(ByteBuffer.wrap(new String(str).getBytes("Utf-8")));
        channel.write(outBuffer);
    }

    /**
     * 处理读取服务端发来的信息 的事件
     *
     * @param key
     * @throws IOException
     */
    public void read(SelectionKey key) throws IOException {
        //和服务端的read方法一样
        //服务器可读取消息:得到事件发生的Socket通道
        SocketChannel channel = (SocketChannel) key.channel();
        // 创建读取的缓冲区
        ByteBuffer buffer = ByteBuffer.allocate(1024);
        int count = channel.read(buffer);

        if (count > 0) {
            buffer.flip();
            CharBuffer charBuffer = Charset.forName("UTF-8").newDecoder().decode(buffer);
            String msg = charBuffer.toString();
            System.out.println("服务端发送消息：" + msg);
        } else {
            channel.close();
        }

        Scanner scanner = new Scanner(System.in);
        System.out.println("请输入：");
        String str = scanner.nextLine();
        if (("exit").equals(str)) {
            isOver = true;
        } else {
            //在这里可以给服务端发送信息哦
            ByteBuffer outBuffer = Charset.forName("UTF-8").newEncoder().encode(CharBuffer.wrap(str));
            channel.write(outBuffer);
        }
    }


    /**
     * 启动客户端测试
     *
     * @throws IOException
     */
    public static void main(String[] args) throws IOException {
        NIOClient client = new NIOClient();
        client.initClient("localhost", 8000);
        client.listen();
    }
}